修改后台编辑窗口
初始后台管理页面中,页面内容仅包括标题项,可通过以下方式添加其他元素。
新增正文编辑框,可编辑 home/models.py :
1 | from django.db import models |
其中,body为一个富文本窗口(特殊wagtail元素),content_panels定义页面容器。
然后运行
python manage.py makemigrations
python manage.py migrate
更新数据库。
每次修改代码之后,都需要运行上述代码,以确保功能更新。
现在,通过后台管理页面->Homepage->Edit,即可看到新的body页面,编辑正文内容,然后发布此页面。
修改前台展示
页面发布之后,返回前台首页,发现仅网页标题改了,其他一切未变,这是因为首页的渲染模板是未变。
编辑 home/templates/home/home_page.html ,使其仅包含以下内容
1 | {% extends "base.html" %} |
再次刷新首页,即可看到已显示编辑内容,但界面好丑….
Wagtail提供很多模板标签,可通过在模板文件头部添加 load wagtailcore_tags 显式引入模板标签。
本文中,使用富文本框来编辑和输出正文内容:
1 | {% load wagtailcore_tags %} |
最终在前台页面,会将其渲染为:
1 | <div class="rich-text"> |
注: 需要在每一个模板文件中包含 load wagtailcore_tags 来使用Wagtail标签,否则Django将抛出错误TemplateSyntaxError。
简单博客网站
在wagtail项目根目录下,通过以下代码,在已有wagtail网站中添加一个blog应用:
python manage.py startapp blog
编辑 mysite/settings/base.py ,在 INSTALLED_APPS 添加 blog
博客首页
通过编辑blog/models.py,定义一个简单的博客首页:
1 | from wagtail.core.models import Page |
运行以下代码,确保更新
python manage.py makemigrations
python manage.py migrate
模板文件名为BlogIndexPage,默认的模板文件位于根目录下/mysite/templates/blog/blog_index_page.html。创建此文件,并包含以下内容:
1 | {% extends "base.html" %} |
在wagtail后台管理界面,创建BlogIndexPage作为Homepage的子页面,确保Promote页签中slug为blog并发布。
现在,可以通过 http://localhost:8000/blog/ 访问你的blog网站。
发布文章页面
编辑 blog/models.py:
1 | from django.db import models |
运行以下代码,确保更新
python manage.py makemigrations
python manage.py migrate
创建模板文件 mysite/templates/blog/blog_page.html:
1 | {% extends "base.html" %} |
可以发布几个博客内容页面,作为博客首页(BlogIndexPage)的子页面,确保创建时选择“Blog Page”类型。
一个较为简陋的博客系统初步搞定。
排序和发布
但是,现在博客首页现在还有一些问题:
- 文章按照时间顺序显示,先发布先显示
- 文章无论是否发布,都显示
变更首页文章排序方式,可通过修改blog/models.py,重写BlogIndexPage的get_context():
1 | class BlogIndexPage(Page): |
修改模板blog_index_page.html,将 ‘for post in page.get_children’ 修改为 ‘for post in blogpages’
现在,试着将一片文章取消发布,即可发现它不再显示在博客首页了。同时,显示顺序也变更为时间倒序。
图片支持
尽管通过富文本框新增图片很容易,但仍然建议将图片作为一个新元素存入数据库,这样的话此图片可独立于当前文章,用在网站任何地方。
在blog/models.py新增BlogPageGalleryImage:
1 | from django.db import models |
运行以下代码,确保更新
python manage.py makemigrations
python manage.py migrate
通过以上步骤,在新增博客文章页面BlogPage增加了一个图片选项。
修正mysite/templates/blog/blog_page.html,使文章显示页面能够显示图片:
1 | {% extends "base.html" %} |
通过在blog/models.py的BlogPage中定义main_image,返回文章中首个图片:
1 | class BlogPage(Page): |
更新 blog_index_page.html ,使博客首页包含图片(首张)的显示:
1 | {% load wagtailcore_tags wagtailimages_tags %} |
标签&分类待后整理
添加标签
首先,修改 blog/models.py:
1 | from django.db import models |
运行以下代码,确保更新
python manage.py makemigrations
python manage.py migrate
Note the new modelcluster and taggit imports, the addition of a new BlogPageTag model, and the addition of a tags field on BlogPage. We’ve also taken the opportunity to use a MultiFieldPanel in content_panels to group the date and tags fields together for readability.
Edit one of your BlogPage instances, and you should now be able to tag posts:
Tagging a post
To render tags on a BlogPage, add this to blog_page.html:
1 | {% if page.tags.all.count %} |
Notice that we’re linking to pages here with the builtin slugurl tag rather than pageurl, which we used earlier. The difference is that slugurl takes a Page slug (from the Promote tab) as an argument. pageurl is more commonly used because it is unambiguous and avoids extra database lookups. But in the case of this loop, the Page object isn’t readily available, so we fall back on the less-preferred slugurl tag.
Visiting a blog post with tags should now show a set of linked buttons at the bottom - one for each tag. However, clicking a button will get you a 404, since we haven’t yet defined a “tags” view. Add to models.py:
1 | class BlogTagIndexPage(Page): |
Note that this Page-based model defines no fields of its own. Even without fields, subclassing Page makes it a part of the Wagtail ecosystem, so that you can give it a title and URL in the admin, and so that you can manipulate its contents by returning a QuerySet from its get_context() method.
Migrate this in, then create a new BlogTagIndexPage in the admin. You’ll probably want to create the new page/view as a child of Homepage, parallel to your Blog index. Give it the slug “tags” on the Promote tab.
Access /tags and Django will tell you what you probably already knew: you need to create a template blog/blog_tag_index_page.html:
1 | {% extends "base.html" %} |
We’re calling the built-in latest_revision_created_at field on the Page model - handy to know this is always available.
We haven’t yet added an “author” field to our BlogPage model, nor do we have a Profile model for authors - we’ll leave those as an exercise for the reader.
Clicking the tag button at the bottom of a BlogPost should now render a page something like this:
A simple tag view
添加分类
Let’s add a category system to our blog. Unlike tags, where a page author can bring a tag into existence simply by using it on a page, our categories will be a fixed list, managed by the site owner through a separate area of the admin interface.
First, we define a BlogCategory model. A category is not a page in its own right, and so we define it as a standard Django models.Model rather than inheriting from Page. Wagtail introduces the concept of “snippets” for reusable pieces of content that need to be managed through the admin interface, but do not exist as part of the page tree themselves; a model can be registered as a snippet by adding the @register_snippet decorator. All the field types we’ve used so far on pages can be used on snippets too - here we’ll give each category an icon image as well as a name. Add to blog/models.py:
1 | from wagtail.snippets.models import register_snippet |
Note
Note that we are using panels rather than content_panels here - since snippets generally have no need for fields such as slug or publish date, the editing interface for them is not split into separate ‘content’ / ‘promote’ / ‘settings’ tabs as standard, and so there is no need to distinguish between ‘content panels’ and ‘promote panels’.
Migrate this change in, and create a few categories through the Snippets area which now appears in the admin menu.
We can now add categories to the BlogPage model, as a many-to-many field. The field type we use for this is ParentalManyToManyField - this is a variant of the standard Django ManyToManyField which ensures that the chosen objects are correctly stored against the page record in the revision history, in much the same way that ParentalKey replaces ForeignKey for one-to-many relations.
1 | # New imports added for forms and ParentalManyToManyField |
Here we’re making use of the widget keyword argument on the FieldPanel definition to specify a checkbox-based widget instead of the default multiple select box, as this is often considered more user-friendly.
Finally, we can update the blog_page.html template to display the categories:
1 | <h1>{{ page.title }}</h1> |